001 /* 002 * Copyright 2004-2005 Stephen McConnell 003 * 004 * Licensed under the Apache License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.apache.org/licenses/LICENSE-2.0 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 013 * implied. 014 * 015 * See the License for the specific language governing permissions and 016 * limitations under the License. 017 */ 018 019 package net.dpml.util; 020 021 import java.io.File; 022 import java.io.FileNotFoundException; 023 import java.io.InputStream; 024 import java.util.ArrayList; 025 import java.util.Properties; 026 027 import javax.xml.parsers.DocumentBuilder; 028 import javax.xml.parsers.DocumentBuilderFactory; 029 030 import org.w3c.dom.Document; 031 import org.w3c.dom.Element; 032 import org.w3c.dom.Node; 033 import org.w3c.dom.NodeList; 034 035 /** 036 * Utility class supporting the translation of DOM content into local child, children, 037 * attribute and value values. 038 * 039 * @author <a href="http://www.dpml.net">The Digital Product Meta Library</a> 040 * @version 1.0.0 041 */ 042 public final class ElementHelper 043 { 044 private ElementHelper() 045 { 046 // utility class 047 } 048 049 /** 050 * Return the root element of the supplied file. 051 * @param definition the file to load 052 * @return the root element 053 * @exception Exception if the error occurs during root element establishment 054 */ 055 public static Element getRootElement( final File definition ) 056 throws Exception 057 { 058 if( !definition.exists() ) 059 { 060 throw new FileNotFoundException( definition.toString() ); 061 } 062 063 if( !definition.isFile() ) 064 { 065 final String error = 066 "Source is not a file: " + definition; 067 throw new IllegalArgumentException( error ); 068 } 069 070 final DocumentBuilderFactory factory = 071 DocumentBuilderFactory.newInstance(); 072 factory.setValidating( false ); 073 factory.setNamespaceAware( false ); 074 final Document document = 075 factory.newDocumentBuilder().parse( definition ); 076 return document.getDocumentElement(); 077 } 078 079 /** 080 * Return the root element of the supplied input stream. 081 * @param input the input stream containing a XML definition 082 * @return the root element 083 * @exception Exception if an error occurs 084 */ 085 public static Element getRootElement( final InputStream input ) 086 throws Exception 087 { 088 final DocumentBuilderFactory factory = 089 DocumentBuilderFactory.newInstance(); 090 factory.setValidating( false ); 091 factory.setNamespaceAware( false ); 092 factory.setExpandEntityReferences( false ); 093 DocumentBuilder builder = factory.newDocumentBuilder(); 094 return getRootElement( builder, input ); 095 } 096 097 /** 098 * Return the root element of the supplied input stream. 099 * @param builder the document builder 100 * @param input the input stream containing a XML definition 101 * @return the root element 102 * @exception Exception if an error occurs 103 */ 104 public static Element getRootElement( final DocumentBuilder builder, final InputStream input ) 105 throws Exception 106 { 107 final Document document = builder.parse( input ); 108 return document.getDocumentElement(); 109 } 110 111 /** 112 * Return a named child relative to a supplied element. 113 * @param root the parent DOM element 114 * @param name the name of a child element 115 * @return the child element of null if the child does not exist 116 */ 117 public static Element getChild( final Element root, final String name ) 118 { 119 if( null == root ) 120 { 121 return null; 122 } 123 Element[] children = getChildren( root ); 124 for( int i=0; i<children.length; i++ ) 125 { 126 Element child = children[i]; 127 if( name.equals( child.getTagName() ) ) 128 { 129 return child; 130 } 131 } 132 return null; 133 } 134 135 /** 136 * Return all children matching the supplied element name. 137 * @param root the parent DOM element 138 * @param name the name against which child element will be matched 139 * @return the array of child elements with a matching name 140 */ 141 public static Element[] getChildren( final Element root, final String name ) 142 { 143 if( null == root ) 144 { 145 return new Element[0]; 146 } 147 Element[] children = getChildren( root ); 148 final ArrayList result = new ArrayList(); 149 for( int i=0; i<children.length; i++ ) 150 { 151 final Element child = children[i]; 152 if( name.equals( child.getTagName() ) ) 153 { 154 result.add( child ); 155 } 156 } 157 return (Element[]) result.toArray( new Element[0] ); 158 } 159 160 /** 161 * Return all children of the supplied parent. 162 * @param root the parent DOM element 163 * @return the array of all children 164 */ 165 public static Element[] getChildren( final Element root ) 166 { 167 if( null == root ) 168 { 169 return new Element[0]; 170 } 171 final NodeList list = root.getChildNodes(); 172 final int n = list.getLength(); 173 if( n < 1 ) 174 { 175 return new Element[0]; 176 } 177 final ArrayList result = new ArrayList(); 178 for( int i=0; i < n; i++ ) 179 { 180 final Node item = list.item( i ); 181 if( item instanceof Element ) 182 { 183 result.add( item ); 184 } 185 } 186 return (Element[]) result.toArray( new Element[0] ); 187 } 188 189 /** 190 * Return the value of an element. 191 * @param node the DOM node 192 * @return the node value 193 */ 194 public static String getValue( final Element node ) 195 { 196 if( null == node ) 197 { 198 return null; 199 } 200 String value; 201 if( node.getChildNodes().getLength() > 0 ) 202 { 203 value = node.getFirstChild().getNodeValue(); 204 } 205 else 206 { 207 value = node.getNodeValue(); 208 } 209 //return normalize( value ); 210 return value; 211 } 212 213 /** 214 * Return the value of an element attribute. 215 * @param node the DOM node 216 * @param key the attribute key 217 * @return the attribute value or null if the attribute is undefined 218 */ 219 public static String getAttribute( final Element node, final String key ) 220 { 221 return getAttribute( node, key, null ); 222 } 223 224 /** 225 * Return the value of an element attribute. 226 * @param node the DOM node 227 * @param key the attribute key 228 * @param def the default value if the attribute is undefined 229 * @return the attribute value or the default value if undefined 230 */ 231 public static String getAttribute( final Element node, final String key, final String def ) 232 { 233 if( null == node ) 234 { 235 return def; 236 } 237 if( !node.hasAttribute( key ) ) 238 { 239 return def; 240 } 241 //final String value = node.getAttribute( key ); 242 //return normalize( value ); 243 return node.getAttribute( key ); 244 } 245 246 /** 247 * Return the value of an element attribute as a boolean 248 * @param node the DOM node 249 * @param key the attribute key 250 * @return the attribute value as a boolean or false if undefined 251 */ 252 public static boolean getBooleanAttribute( final Element node, final String key ) 253 { 254 return getBooleanAttribute( node, key, false ); 255 } 256 257 /** 258 * Return the value of an element attribute as a boolean. 259 * @param node the DOM node 260 * @param key the attribute key 261 * @param def the default value if the attribute is undefined 262 * @return the attribute value or the default value if undefined 263 */ 264 public static boolean getBooleanAttribute( final Element node, final String key, final boolean def ) 265 { 266 if( null == node ) 267 { 268 return def; 269 } 270 271 if( !node.hasAttribute( key ) ) 272 { 273 return def; 274 } 275 276 String value = node.getAttribute( key ); 277 value = normalize( value ); 278 if( value.equals( "" ) ) 279 { 280 return def; 281 } 282 if( value.equalsIgnoreCase( "true" ) ) 283 { 284 return true; 285 } 286 if( value.equalsIgnoreCase( "false" ) ) 287 { 288 return false; 289 } 290 final String error = 291 "Boolean argument [" + value + "] not recognized."; 292 throw new IllegalArgumentException( error ); 293 } 294 295 /** 296 * Parse the value for any property tokens relative to system properties. 297 * @param value the value to parse 298 * @return the normalized string 299 */ 300 static String normalize( String value ) 301 { 302 return normalize( value, System.getProperties() ); 303 } 304 305 /** 306 * Parse the value for any property tokens relative to the supplied properties. 307 * @param value the value to parse 308 * @param props the reference properties 309 * @return the normalized string 310 */ 311 static String normalize( String value, Properties props ) 312 { 313 return PropertyResolver.resolve( props, value ); 314 } 315 }